Android Recyclerview间距 均分 完美布局 支持任意列数和两端间距

最近项目中要在RecyclerView的Grid中均分间距,看了源码和一些文章,下面的这篇很有启发,在此引用一下,并且在他的基础上支持任意列数和两端间距。最后附上代码:

原文链接:https://blog.csdn.net/lovext4098477/article/details/80419201

简介:

  App中,用到最多的设计就是列表形式的布局,而RecyclerView的出现,也完完全全的替代了之前的Listview、GridView,成为android控件中,用途最为广泛的widget之一,今天就来简单介绍一下,RecyclerView的一些布局方法。

实战:

  我们经常在设计稿中看到各式各样的列表,最通常的需求,就是对各个item进行排列,这就运用到了对分割线的处理。在RecyclerView中,有一个public void addItemDecoration(RecyclerView.ItemDecoration decor)的方法,RecyclerView.ItemDecoration这个类里包含了一个getItemOffsets的方法,我们就是要通过这个方法去对每个item分割,通过设置不同的offset来改变间距。以下图这个相册的为例:

 

  让我们来简单地剖析一下布局,假设屏幕是720px,每个间距为20px,通过计算,每个item的长宽为165px。

  这时候,很多人会考虑这样去设置

  我这里是dp转px,720是2x,所以delta是10px,通过parent.getChildAdapterPosition(view)来判断是每一行的第几个,然后第一个item设置了左0右10,第二个左10右10,第三个10右10,第四个左10右0。嗯,理论上来看好像是这样的,动手试试。结果发现,这样的布局并没有到达上图设计的效果。这是为什么呢?

  首先,我们要了解RecyclerView的分割原理,当一个RecyclerView设置了一个GridLayoutManager(this,count),并且count为4的时候,实际上就是将屏幕均分为四份,每一份都是180px宽(以720px为例,我们只考虑左右,暂不考虑上下,原理是相同的),如果不设置ItemDecoration,那么默认item由左开始布置,也就是说,165px的View在它的布局中是这样婶儿的。

  显而易见,无论我如何设置,item距边界最多就15px(left+right),如果要保证第一个item和第二个item间距为20px,那么我只需要将第二个item设置一个left为5px就可以达到想要的效果。第三第四个也同理。

  所以正确的左右设置应该为

 

在此基础上进行改进,支持任意列数和两端间距:

核心代码:

RecyclerView.LayoutManager rvLayoutManager = parent.getLayoutManager();

        if (rvLayoutManager == null || parent.getAdapter() == null) {
            return;
        }
        if (gapHSizePx < 0 || gapVSizePx < 0) {

            DisplayMetrics displayMetrics = new DisplayMetrics();
            parent.getDisplay().getMetrics(displayMetrics);
            gapHSizePx = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, gapHorizontalDp, displayMetrics);
            gapVSizePx = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, gapVerticalDp, displayMetrics);
            edgePaddingPx = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, edgePaddingDp, displayMetrics);

            if (rvLayoutManager instanceof GridLayoutManager) {
                GridLayoutManager layoutManager = (GridLayoutManager) parent.getLayoutManager();
                spanCount = layoutManager.getSpanCount();
                eachItemPaddingH = (edgePaddingPx * 2 + gapHSizePx * (spanCount - 1)) / spanCount;
            } else if (rvLayoutManager instanceof StaggeredGridLayoutManager) {
                StaggeredGridLayoutManager layoutManager = (StaggeredGridLayoutManager) parent.getLayoutManager();
                spanCount = layoutManager.getSpanCount();
                eachItemPaddingH = (gapHSizePx * (spanCount - 1)) / spanCount;
                //特殊设置
                int rvPadding = 0;
                if (layoutManager.getOrientation() == StaggeredGridLayoutManager.VERTICAL) {
                    rvPadding = edgePaddingPx - gapHSizePx / 2;
                } else {
                    rvPadding = edgePaddingPx - gapVSizePx / 2;
                }
                rvPadding = rvPadding >= 0 ? rvPadding : 0;
                parent.setPadding(rvPadding, 0, rvPadding, 0);
            } else if (rvLayoutManager instanceof LinearLayoutManager) {
                spanCount = 1;
                eachItemPaddingH = (edgePaddingPx * 2 + gapHSizePx * (spanCount - 1)) / spanCount;
            } else {
                throw new RuntimeException("不支持的LayoutManager");
            }
        }

        int position = parent.getChildAdapterPosition(view);
        if (!handleDecorationAtAdapterPosition(outRect, view, parent, state, position)) {
            if (rvLayoutManager instanceof GridLayoutManager) {
                setGridDecoration(outRect, view, parent, state, position);
            } else if (rvLayoutManager instanceof StaggeredGridLayoutManager) {
                setStaggeredGridDecoration((StaggeredGridLayoutManager) rvLayoutManager, outRect, view, parent, state, position);
            } else if (rvLayoutManager instanceof LinearLayoutManager) {
                spanCount = 1;
                setLinearDecoration((LinearLayoutManager) rvLayoutManager, outRect, view, parent, state, position);
            } else {
                throw new RuntimeException("不支持的LayoutManager");
            }
        }
        if (BuildConfig.DEBUG) {
            Log.d("ItemDecoration", "pos=" + position + "," + outRect.toShortString());
        }

下载地址:GridAverageGapItemDecoration

另外如果项目中用到了Brvah,并且想自定义Section中的间距的话,看这篇:

BRVAH的Section中自定义间距,条目完美均分,支持Section顶部和底部间距

  • 3
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Android 开发中,可以通过设置 RecyclerView间距来调整其显示效果。下面是一些实现方法: 1. 在 RecyclerView 的 item 布局文件中添加 margin 属性。例如,在 LinearLayout 布局中,可以通过设置 android:layout_marginTop、android:layout_marginBottom、android:layout_marginLeft、android:layout_marginRight 属性来设置上下左右的间距。 2. 通过 RecyclerView.ItemDecoration 实现间距设置。自定义一个 RecyclerView.ItemDecoration 类,重写 onDraw() 和 getItemOffsets() 方法,在 getItemOffsets() 方法中设置 item 之间的间距。 ```java public class MyItemDecoration extends RecyclerView.ItemDecoration { private int space; public MyItemDecoration(int space) { this.space = space; } @Override public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) { outRect.left = space; outRect.right = space; outRect.bottom = space; if (parent.getChildLayoutPosition(view) == 0) { outRect.top = space; } else { outRect.top = 0; } } } ``` 在 RecyclerView 的代码中调用 setItemDecoration() 方法,将自定义的 MyItemDecoration 类传入即可。 ```java recyclerView.addItemDecoration(new MyItemDecoration(10)); ``` 3. 使用 RecyclerView 的 LayoutManager 来实现间距设置。例如,使用 LinearLayoutManager 的 setPadding() 和 setClipToPadding() 方法来设置间距。 ```java LinearLayoutManager layoutManager = new LinearLayoutManager(this); recyclerView.setLayoutManager(layoutManager); recyclerView.setPadding(10, 10, 10, 10); recyclerView.setClipToPadding(false); ``` 以上是三种常用的设置 RecyclerView 间距的方法,可以根据具体需求选择适合自己的方式。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值